#ifndef __SELECT_SVR_H__
#define __SELECT_SVR_H__

#include <iostream>
#include <string>
#include <vector>
#include <sys/select.h>
#include <sys/time.h>
#include "Log.hpp"
#include "Sock.hpp"

using namespace std;

#define BITS 8
#define NUM (sizeof(fd_set) * BITS)
#define FD_NONE -1

class SelectServer
{
public:
    SelectServer(const uint16_t &port = 8080)
        : _port(port)
    {
        _listensock = Sock::Socket();
        Sock::Bind(_listensock, _port);
        Sock::Listen(_listensock);
        logMessage(DEBUG, "%s", "create base socket success");
        for (int i = 0; i < NUM; i++)
            _fd_array[i] = FD_NONE;
        // 规定_fd_array[0] = _listensock;
        _fd_array[0] = _listensock;
    }

    void Start()
    {
        while (true)
        {
            // struct timeval timeout = {0, 0};
            // 如何看待listensock？获取新链接，我们把它依旧看做成为IO，input事件，如果没有链接到来？不就阻塞了吗？
            // int sock = Sock::Accept(_listensock, ...); // 不能直接调用accept了
            // FD_SET(_listensock, &rfds); // 将listensock添加到读文件描述符集中
            // int n = select(_listensock + 1, &rfds, nullptr, nullptr, &timeout);

            // 1. nfds：随着获取的sock越来越多，随着添加到select的sock越来越多，注定了nfds每一次都可能要变化，需要对它动态计算
            // 2. rfds/writefds/excepfds: 都是输入输出型参数，输入输出不一定是一样的，所以注定了每一次都要对rfds进行重新添加
            // 3. timeout：都是输入输出型参数，每一次都要进行重置，前提是你要的话
            // 1,2=> 注定必须自己将合法的文件描述符需要单独保存起来，用来支持：1.更新最大fd 2.更新位图结构

            DebugPrint();

            fd_set rfds;
            FD_ZERO(&rfds);
            int maxfd = _listensock;
            for (int i = 0; i < NUM; i++)
            {
                if (_fd_array[i] == FD_NONE)
                    continue;
                FD_SET(_fd_array[i], &rfds);
                if (maxfd < _fd_array[i])
                    maxfd = _fd_array[i];
            }

            // rfds未来一定会有两类sock，listensock，普通sock
            // select中，就绪的fd会越来越多！
            int n = select(maxfd + 1, &rfds, nullptr, nullptr, nullptr);
            switch (n)
            {
            case 0:
                logMessage(DEBUG, "%s", "time out...");
                break;
            case -1:
                logMessage(WARNING, "select error: %d : %s", errno, strerror(errno));
                break;
            default:
                // 成功的
                logMessage(DEBUG, "get a new link event..."); // 为什么会一直打印链接到来呢？链接已经建立完成，就绪了，但是没有取走，select要一直通知你
                HandlerEvent(rfds);
                break;
            }
        }
    }

    ~SelectServer()
    {
        if (_listensock >= 0)
            close(_listensock);
    }

private:
    void HandlerEvent(const fd_set &rfds) // fd_set是一个集合，里面可能会存在多个sock
    {
        for (int i = 0; i < NUM; i++)
        {
            // 1. 去掉不合法的fd
            if (_fd_array[i] == FD_NONE)
                continue;
            // 2. 合法的一定是就绪的吗
            if (FD_ISSET(_fd_array[i], &rfds))
            {
                // 指定的fd，读事件就绪，就一定是read吗
                if (_fd_array[i] == _listensock)
                {
                    // 读事件就绪：链接事件到来，accept
                    Accepter();
                }
                else
                {
                    Recver(i);
                }
            }
        }
    }
    void Accepter()
    {
        string clientip;
        uint16_t clientport = 0;

        // listensock上面的读事件就绪了，表示可以读取了
        // 获取新链接
        int sock = Sock::Accept(_listensock, &clientip, &clientport); // 这里进行accept会不会阻塞？不会
        if (sock < 0)
        {
            logMessage(WARNING, "accept error");
            return;
        }
        logMessage(DEBUG, "get a new line success : [%s:%d]", clientip.c_str(), clientport, sock);

        // read / recv? 不能，因为不清楚sock上面的数据什么时候来，recv、readjiu有可能先被阻塞 IO=等+数据拷贝
        // select最清楚
        // 得到了新链接的时候，此时应该将新的sock托管给select，让select帮我们进行检测sock上是否有新的数据
        // 有了数据select，读事件就绪，select就会通知我，再进行读取，此时就不会被阻塞了
        // 要将sock添加给select，其实只要将fd放入到数组中即可
        int pos = 1;
        for (; pos < NUM; pos++)
        {
            if (_fd_array[pos] == FD_NONE)
                break;
        }
        if (pos == NUM)
        {
            logMessage(WARNING, "%s:%d", "select server already full, close: %d", sock);
            close(sock);
        }
        else
        {
            _fd_array[pos] = sock;
        }
    }
    void Recver(int pos)
    {
        // 读事件就绪：INPUT事件到来，recv，read
        logMessage(DEBUG, "message in, get IO event: %d", _fd_array[pos]);
        // 此时select已经进行了事件检测，fd上的数据一定是就绪的，即本次不会阻塞
        // 这里是有bug的，你怎么保证可以读到一个完整的报文呢
        char buffer[1024];
        int n = recv(_fd_array[pos], buffer, sizeof(buffer) - 1, 0);
        if (n > 0)
        {
            buffer[n] = 0;
            logMessage(DEBUG, "client[%d]# %s", _fd_array[pos], buffer);
        }
        else if (n == 0)
        {
            logMessage(DEBUG, "client[%d] quit, me too...", _fd_array[pos]);
            // 1. 我们也要关闭不需要的fd
            close(_fd_array[pos]);
            // 2. 不要让select帮我关心当前的fd了
            _fd_array[pos] = FD_NONE;
        }
        else
        {
            logMessage(WARNING, "%d sock recv error, %d : %s", _fd_array[pos], errno, strerror(errno));
            // 1. 我们也要关闭不需要的fd
            close(_fd_array[pos]);
            // 2. 不要让select帮我关心当前的fd了
            _fd_array[pos] = FD_NONE;
        }
    }
    void DebugPrint()
    {
        cout << "_fd_array[]: ";
        for (int i = 0; i < NUM; i++)
        {
            if (_fd_array[i] == FD_NONE)
                continue;
            cout << _fd_array[i] << " ";
        }
        cout << endl;
    }

private:
    uint16_t _port;
    int _listensock;
    int _fd_array[NUM];
    // int _fd_read[NUM];
    // int _fd_write[NUM];
};

#endif